package main

import (
	"context"
	"fmt"
	"net/http"
	"os"
	"os/signal"
	"syscall"
	"time"

	"go.uber.org/zap"

	"bluebell/dao/mysql"
	"bluebell/models"
	"bluebell/pkg/snowflake"

	// "bluebell/dao/redis"
	"bluebell/logger"
	"bluebell/router"
	"bluebell/setting"
)

func main() {
	// 1.加载配置文件
	if err := setting.Init();err != nil{
		fmt.Printf("setting.Init set err:%s",err)
	}
	// 2.初始化日志
	if err := logger.Init();err != nil{
		fmt.Printf("logger.Init set err:%s",err)
	}
	defer zap.L().Sync()
	// 3.初始化mysql
	if err := mysql.Init();err != nil{
		fmt.Printf("mysql.Init set err:%s",err)
	}
	mysql.DB.AutoMigrate(&models.User{})
	if err := snowflake.Init(setting.Conf.StartTime,setting.Conf.MachineId);err != nil{
		fmt.Printf("snowflake.Init set err:%s",err)
	}
	fmt.Println(snowflake.GetID())
	// 4.初始化redis
	// if err := redis.Init();err != nil{
	// 	fmt.Printf("setting.Init set err:%s",err)
	// }
	// 5.注册路由
	r := router.SetUp();
	// 6.优雅启停
	srv := &http.Server{
		Addr:    setting.Conf.Port,
		Handler: r,
	}

	go func() {
		// 开启一个goroutine启动服务
		if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
			zap.L().Error("listen:", zap.Error(err))
		}
	}()

	// 等待中断信号来优雅地关闭服务器，为关闭服务器操作设置一个5秒的超时
	quit := make(chan os.Signal, 1) // 创建一个接收信号的通道
	// kill 默认会发送 syscall.SIGTERM 信号
	// kill -2 发送 syscall.SIGINT 信号，我们常用的Ctrl+C就是触发系统SIGINT信号
	// kill -9 发送 syscall.SIGKILL 信号，但是不能被捕获，所以不需要添加它
	// signal.Notify把收到的 syscall.SIGINT或syscall.SIGTERM 信号转发给quit
	signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)  // 此处不会阻塞
	<-quit  // 阻塞在此，当接收到上述两种信号时才会往下执行
	zap.L().Info("Shutdown Server ...")
	// 创建一个5秒超时的context
	ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
	defer cancel()
	// 5秒内优雅关闭服务（将未处理完的请求处理完再关闭服务），超过5秒就超时退出
	if err := srv.Shutdown(ctx); err != nil {
		zap.L().Error("Server Shutdown",zap.Error(err))
	}
	zap.L().Info("Server exiting")
}